home *** CD-ROM | disk | FTP | other *** search
- /* Copyright (C) 1996, 1997, 1998, 1999 Aladdin Enterprises. All rights reserved.
-
- This file is part of AFPL Ghostscript.
-
- AFPL Ghostscript is distributed with NO WARRANTY OF ANY KIND. No author or
- distributor accepts any responsibility for the consequences of using it, or
- for whether it serves any particular purpose or works at all, unless he or
- she says so in writing. Refer to the Aladdin Free Public License (the
- "License") for full details.
-
- Every copy of AFPL Ghostscript must include a copy of the License, normally
- in a plain ASCII text file named PUBLIC. The License grants you the right
- to copy, modify and redistribute AFPL Ghostscript, but only under certain
- conditions described in the License. Among other things, the License
- requires that the copyright notice and this notice be preserved on all
- copies.
- */
-
- /*$Id: gxclimag.c,v 1.2 2000/09/19 19:00:34 lpd Exp $ */
- /* Higher-level image operations for band lists */
- #include "math_.h"
- #include "memory_.h"
- #include "gx.h"
- #include "gserrors.h"
- #include "gscspace.h"
- #include "gscdefs.h" /* for image type table */
- #include "gxarith.h"
- #include "gxcspace.h"
- #include "gxdevice.h"
- #include "gxdevmem.h" /* must precede gxcldev.h */
- #include "gxcldev.h"
- #include "gxclpath.h"
- #include "gxfmap.h"
- #include "gxiparam.h"
- #include "gxpath.h"
- #include "stream.h"
- #include "strimpl.h" /* for sisparam.h */
- #include "sisparam.h"
-
- extern_gx_image_type_table();
-
- /* Define whether we should use high-level images. */
- /* (See below for additional restrictions.) */
- static bool USE_HL_IMAGES = true;
-
- /* Forward references */
- private int cmd_put_set_data_x(P3(gx_device_clist_writer * cldev,
- gx_clist_state * pcls, int data_x));
- private int cmd_put_color_mapping(P3(gx_device_clist_writer * cldev,
- const gs_imager_state * pis,
- bool write_rgb_to_cmyk));
- private bool check_rect_for_trivial_clip(P5(
- const gx_clip_path *pcpath, /* May be NULL, clip to evaluate */
- int px, int py, int qx, int qy /* corners of box to test */
- ));
-
- /* ------ Driver procedures ------ */
-
- int
- clist_fill_mask(gx_device * dev,
- const byte * data, int data_x, int raster, gx_bitmap_id id,
- int x, int y, int width, int height,
- const gx_drawing_color * pdcolor, int depth,
- gs_logical_operation_t lop, const gx_clip_path * pcpath)
- {
- gx_device_clist_writer * const cdev =
- &((gx_device_clist *)dev)->writer;
- const byte *orig_data = data; /* for writing tile */
- int orig_data_x = data_x; /* ditto */
- int orig_x = x; /* ditto */
- int orig_width = width; /* ditto */
- int orig_height = height; /* ditto */
- int log2_depth = ilog2(depth);
- int y0;
- int data_x_bit;
- byte copy_op =
- (depth > 1 ? cmd_op_copy_color_alpha :
- gx_dc_is_pure(pdcolor) ? cmd_op_copy_mono :
- cmd_op_copy_mono + cmd_copy_ht_color);
- bool slow_rop =
- cmd_slow_rop(dev, lop_know_S_0(lop), pdcolor) ||
- cmd_slow_rop(dev, lop_know_S_1(lop), pdcolor);
- fit_copy(dev, data, data_x, raster, id, x, y, width, height);
- y0 = y; /* must do after fit_copy */
-
- /* If non-trivial clipping & complex clipping disabled, default */
- if (((cdev->disable_mask & clist_disable_complex_clip) &&
- !check_rect_for_trivial_clip(pcpath, x, y, x + width, y + height)) ||
- gs_debug_c('`')
- )
- return gx_default_fill_mask(dev, data, data_x, raster, id,
- x, y, width, height, pdcolor, depth,
- lop, pcpath);
- if (cmd_check_clip_path(cdev, pcpath))
- cmd_clear_known(cdev, clip_path_known);
- data_x_bit = data_x << log2_depth;
- FOR_RECTS {
- int dx = (data_x_bit & 7) >> log2_depth;
- const byte *row = data + (y - y0) * raster + (data_x_bit >> 3);
- int code;
-
- TRY_RECT {
- code = cmd_update_lop(cdev, pcls, lop);
- } HANDLE_RECT(code);
- if (depth > 1 && !pcls->color_is_alpha) {
- byte *dp;
-
- TRY_RECT {
- code =
- set_cmd_put_op(dp, cdev, pcls, cmd_opv_set_copy_alpha, 1);
- } HANDLE_RECT(code);
- pcls->color_is_alpha = 1;
- }
- TRY_RECT {
- code = cmd_do_write_unknown(cdev, pcls, clip_path_known);
- if (code >= 0)
- code = cmd_do_enable_clip(cdev, pcls, pcpath != NULL);
- } HANDLE_RECT(code);
- TRY_RECT {
- code = cmd_put_drawing_color(cdev, pcls, pdcolor);
- } HANDLE_RECT(code);
- pcls->colors_used.slow_rop |= slow_rop;
- /*
- * Unfortunately, painting a character with a halftone requires the
- * use of two bitmaps, a situation that we can neither represent in
- * the band list nor guarantee will both be present in the tile
- * cache; in this case, we always write the bits of the character.
- *
- * We could handle more RasterOp cases here directly, but it
- * doesn't seem worth the trouble right now.
- */
- if (id != gx_no_bitmap_id && gx_dc_is_pure(pdcolor) &&
- lop == lop_default
- ) { /* This is a character. ****** WRONG IF HALFTONE CELL. ***** */
- /* Put it in the cache if possible. */
- ulong offset_temp;
-
- if (!cls_has_tile_id(cdev, pcls, id, offset_temp)) {
- gx_strip_bitmap tile;
-
- tile.data = (byte *) orig_data; /* actually const */
- tile.raster = raster;
- tile.size.x = tile.rep_width = orig_width;
- tile.size.y = tile.rep_height = orig_height;
- tile.rep_shift = tile.shift = 0;
- tile.id = id;
- TRY_RECT {
- code = clist_change_bits(cdev, pcls, &tile, depth);
- } HANDLE_RECT_UNLESS(code,
- (code != gs_error_VMerror || !cdev->error_is_retryable) );
- if (code < 0) {
- /* Something went wrong; just copy the bits. */
- goto copy;
- }
- } {
- gx_cmd_rect rect;
- int rsize;
- byte op = copy_op + cmd_copy_use_tile;
-
- /* Output a command to copy the entire character. */
- /* It will be truncated properly per band. */
- rect.x = orig_x, rect.y = y0;
- rect.width = orig_width, rect.height = yend - y0;
- rsize = 1 + cmd_sizexy(rect);
- TRY_RECT {
- code = (orig_data_x ?
- cmd_put_set_data_x(cdev, pcls, orig_data_x) : 0);
- if (code >= 0) {
- byte *dp;
-
- code = set_cmd_put_op(dp, cdev, pcls, op, rsize);
- /*
- * The following conditional is unnecessary: the two
- * statements inside it should go outside the
- * HANDLE_RECT. They are here solely to pacify
- * stupid compilers that don't understand that dp
- * will always be set if control gets past the
- * HANDLE_RECT.
- */
- if (code >= 0) {
- dp++;
- cmd_putxy(rect, dp);
- }
- }
- } HANDLE_RECT(code);
- pcls->rect = rect;
- goto end;
- }
- }
- copy: /*
- * The default fill_mask implementation uses strip_copy_rop;
- * this is exactly what we want.
- */
- TRY_RECT {
- NEST_RECT {
- code = gx_default_fill_mask(dev, row, dx, raster,
- (y == y0 && height == orig_height &&
- dx == orig_data_x ? id :
- gx_no_bitmap_id),
- x, y, width, height, pdcolor,
- depth, lop, pcpath);
- } UNNEST_RECT;
- } HANDLE_RECT(code);
- end:
- ;
- } END_RECTS;
- return 0;
- }
-
- /* ------ Bitmap image driver procedures ------ */
-
- /* Define the structure for keeping track of progress through an image. */
- typedef struct clist_image_enum_s {
- gx_image_enum_common;
- /* Arguments of begin_image */
- gs_memory_t *memory;
- gs_pixel_image_t image; /* only uses Width, Height, Interpolate */
- gx_drawing_color dcolor; /* only pure right now */
- gs_int_rect rect;
- const gs_imager_state *pis;
- const gx_clip_path *pcpath;
- /* Set at creation time */
- gs_image_format_t format;
- gs_int_point support; /* extra source pixels for interpolation */
- int bits_per_plane; /* bits per pixel per plane */
- gs_matrix matrix; /* image space -> device space */
- bool uses_color;
- clist_color_space_t color_space;
- int ymin, ymax;
- bool map_rgb_to_cmyk;
- gx_colors_used_t colors_used;
- /* begin_image command prepared & ready to output */
- /****** SIZE COMPUTATION IS WRONG, TIED TO gximage.c, gsmatrix.c ******/
- byte begin_image_command[3 +
- /* Width, Height */
- 2 * cmd_sizew_max +
- /* ImageMatrix */
- 1 + 6 * sizeof(float) +
- /* Decode */
- (GS_IMAGE_MAX_COMPONENTS + 3) / 4 +
- GS_IMAGE_MAX_COMPONENTS * 2 * sizeof(float) +
- /* MaskColors */
- GS_IMAGE_MAX_COMPONENTS * cmd_sizew_max +
- /* rect */
- 4 * cmd_sizew_max];
- int begin_image_command_length;
- /* Updated dynamically */
- int y;
- bool color_map_is_known;
- } clist_image_enum;
- gs_private_st_suffix_add3(st_clist_image_enum, clist_image_enum,
- "clist_image_enum", clist_image_enum_enum_ptrs,
- clist_image_enum_reloc_ptrs, st_gx_image_enum_common,
- pis, pcpath, color_space.space);
-
- private image_enum_proc_plane_data(clist_image_plane_data);
- private image_enum_proc_end_image(clist_image_end_image);
- private const gx_image_enum_procs_t clist_image_enum_procs =
- {
- clist_image_plane_data, clist_image_end_image
- };
-
- /* Forward declarations */
- private bool image_band_box(P5(gx_device * dev, const clist_image_enum * pie,
- int y, int h, gs_int_rect * pbox));
- private int begin_image_command(P3(byte *buf, uint buf_size,
- const gs_image_common_t *pic));
- private int cmd_image_plane_data(P8(gx_device_clist_writer * cldev,
- gx_clist_state * pcls,
- const gx_image_plane_t * planes,
- const gx_image_enum_common_t * pie,
- uint bytes_per_plane,
- const uint * offsets, int dx, int h));
- private uint clist_image_unknowns(P2(gx_device *dev,
- const clist_image_enum *pie));
- private int write_image_end_all(P2(gx_device *dev,
- const clist_image_enum *pie));
-
- /*
- * Since currently we are limited to writing a single subrectangle of the
- * image for each band, images that are rotated by angles other than
- * multiples of 90 degrees may wind up writing many copies of the data.
- * Eventually we will fix this by breaking up the image into multiple
- * subrectangles, but for now, don't use the high-level approach if it would
- * cause the data to explode because of this.
- */
- private bool
- image_matrix_ok_to_band(const gs_matrix * pmat)
- {
- double t;
-
- /* Don't band if the matrix is (nearly) singular. */
- if (fabs(pmat->xx * pmat->yy - pmat->xy * pmat->yx) < 0.001)
- return false;
- if (is_xxyy(pmat) || is_xyyx(pmat))
- return true;
- t = (fabs(pmat->xx) + fabs(pmat->yy)) /
- (fabs(pmat->xy) + fabs(pmat->yx));
- return (t < 0.2 || t > 5);
- }
-
- /* Start processing an image. */
- int
- clist_begin_typed_image(gx_device * dev,
- const gs_imager_state * pis, const gs_matrix * pmat,
- const gs_image_common_t * pic, const gs_int_rect * prect,
- const gx_drawing_color * pdcolor, const gx_clip_path * pcpath,
- gs_memory_t * mem, gx_image_enum_common_t ** pinfo)
- {
- const gs_pixel_image_t * const pim = (const gs_pixel_image_t *)pic;
- gx_device_clist_writer * const cdev =
- &((gx_device_clist *)dev)->writer;
- clist_image_enum *pie = 0;
- int base_index;
- bool indexed;
- bool masked = false;
- bool has_alpha = false;
- int num_components;
- int bits_per_pixel;
- bool uses_color;
- bool varying_depths = false;
- gs_matrix mat;
- gs_rect sbox, dbox;
- gs_image_format_t format;
- gx_color_index colors_used = 0;
- int code;
-
- /* We can only handle a limited set of image types. */
- switch ((gs_debug_c('`') ? -1 : pic->type->index)) {
- case 1:
- masked = ((const gs_image1_t *)pim)->ImageMask;
- has_alpha = ((const gs_image1_t *)pim)->Alpha != 0;
- case 4:
- if (pmat == 0)
- break;
- default:
- goto use_default;
- }
- format = pim->format;
- /* See above for why we allocate the enumerator as immovable. */
- pie = gs_alloc_struct_immovable(mem, clist_image_enum,
- &st_clist_image_enum,
- "clist_begin_typed_image");
- if (pie == 0)
- return_error(gs_error_VMerror);
- pie->memory = mem;
- *pinfo = (gx_image_enum_common_t *) pie;
- /* num_planes and plane_depths[] are set later, */
- /* by gx_image_enum_common_init. */
- if (masked) {
- base_index = gs_color_space_index_DeviceGray; /* arbitrary */
- indexed = false;
- num_components = 1;
- uses_color = true;
- /* cmd_put_drawing_color handles colors_used */
- } else {
- const gs_color_space *pcs = pim->ColorSpace;
-
- base_index = gs_color_space_get_index(pcs);
- if (base_index == gs_color_space_index_Indexed) {
- const gs_color_space *pbcs =
- gs_color_space_indexed_base_space(pcs);
-
- indexed = true;
- base_index = gs_color_space_get_index(pbcs);
- num_components = 1;
- } else {
- indexed = false;
- num_components = gs_color_space_num_components(pcs);
- }
- uses_color = pim->CombineWithColor && rop3_uses_T(pis->log_op);
- }
- code = gx_image_enum_common_init((gx_image_enum_common_t *) pie,
- (const gs_data_image_t *) pim,
- &clist_image_enum_procs, dev,
- num_components, format);
- {
- int i;
-
- for (i = 1; i < pie->num_planes; ++i)
- varying_depths |= pie->plane_depths[i] != pie->plane_depths[0];
- }
- if (code < 0 ||
- !USE_HL_IMAGES || /* Always use the default. */
- (cdev->disable_mask & clist_disable_hl_image) ||
- cdev->image_enum_id != gs_no_id || /* Can't handle nested images */
- /****** CAN'T HANDLE CIE COLOR YET ******/
- base_index > gs_color_space_index_DeviceCMYK ||
- /****** CAN'T HANDLE NON-PURE COLORS YET ******/
- (uses_color && !gx_dc_is_pure(pdcolor)) ||
- /****** CAN'T HANDLE IMAGES WITH ALPHA YET ******/
- has_alpha ||
- /****** CAN'T HANDLE IMAGES WITH IRREGULAR DEPTHS ******/
- varying_depths ||
- (code = gs_matrix_invert(&pim->ImageMatrix, &mat)) < 0 ||
- (code = gs_matrix_multiply(&mat, &ctm_only(pis), &mat)) < 0 ||
- !(cdev->disable_mask & clist_disable_nonrect_hl_image ?
- (is_xxyy(&mat) || is_xyyx(&mat)) :
- image_matrix_ok_to_band(&mat))
- )
- goto use_default;
- {
- int bytes_per_plane, bytes_per_row;
-
- bits_per_pixel = pim->BitsPerComponent * num_components;
- pie->image = *pim;
- pie->dcolor = *pdcolor;
- if (prect)
- pie->rect = *prect;
- else {
- pie->rect.p.x = 0, pie->rect.p.y = 0;
- pie->rect.q.x = pim->Width, pie->rect.q.y = pim->Height;
- }
- pie->pis = pis;
- pie->pcpath = pcpath;
- pie->format = format;
- pie->bits_per_plane = bits_per_pixel / pie->num_planes;
- pie->matrix = mat;
- pie->uses_color = uses_color;
- if (masked) {
- pie->color_space.byte1 = 0; /* arbitrary */
- pie->color_space.space = 0;
- pie->color_space.id = gs_no_id;
- } else {
- pie->color_space.byte1 = (base_index << 4) |
- (indexed ? (pim->ColorSpace->params.indexed.use_proc ? 12 : 8) : 0);
- pie->color_space.id =
- (pie->color_space.space = pim->ColorSpace)->id;
- }
- pie->y = pie->rect.p.y;
-
- /* Image row has to fit in cmd writer's buffer */
- bytes_per_plane =
- (pim->Width * pie->bits_per_plane + 7) >> 3;
- bytes_per_row = bytes_per_plane * pie->num_planes;
- bytes_per_row = max(bytes_per_row, 1);
- if (cmd_largest_size + bytes_per_row > cdev->cend - cdev->cbuf)
- goto use_default;
- }
- if (pim->Interpolate)
- pie->support.x = pie->support.y = MAX_ISCALE_SUPPORT + 1;
- else
- pie->support.x = pie->support.y = 0;
- sbox.p.x = pie->rect.p.x - pie->support.x;
- sbox.p.y = pie->rect.p.y - pie->support.y;
- sbox.q.x = pie->rect.q.x + pie->support.x;
- sbox.q.y = pie->rect.q.y + pie->support.y;
- gs_bbox_transform(&sbox, &mat, &dbox);
-
- if (cdev->disable_mask & clist_disable_complex_clip)
- if (!check_rect_for_trivial_clip(pcpath,
- (int)floor(dbox.p.x), (int)floor(dbox.p.y),
- (int)ceil(dbox.q.x), (int)ceil(dbox.q.y)))
- goto use_default;
- /* Create the begin_image command. */
- if ((pie->begin_image_command_length =
- begin_image_command(pie->begin_image_command,
- sizeof(pie->begin_image_command), pic)) < 0)
- goto use_default;
- if (!masked) {
- /*
- * Calculate (conservatively) the set of colors that this image
- * might generate. For single-component images with up to 4 bits
- * per pixel, standard Decode values, and no Interpolate, we
- * generate all the possible colors now; otherwise, we assume that
- * any color might be generated. It is possible to do better than
- * this, but we won't bother unless there's evidence that it's
- * worthwhile.
- */
- gx_color_index all =
- ((gx_color_index)1 << dev->color_info.depth) - 1;
-
- if (bits_per_pixel > 4 || pim->Interpolate || num_components > 1)
- colors_used = all;
- else {
- int max_value = (1 << bits_per_pixel) - 1;
- float dmin = pim->Decode[0], dmax = pim->Decode[1];
- float dtemp;
-
- if (dmax < dmin)
- dtemp = dmax, dmax = dmin, dmin = dtemp;
- if (dmin != 0 ||
- dmax != (indexed ? max_value : 1)
- ) {
- colors_used = all;
- } else {
- /* Enumerate the possible pixel values. */
- const gs_color_space *pcs = pim->ColorSpace;
- cs_proc_remap_color((*remap_color)) = pcs->type->remap_color;
- gs_client_color cc;
- gx_drawing_color dcolor;
- int i;
- double denom = (indexed ? 1 : max_value);
-
- for (i = 0; i <= max_value; ++i) {
- cc.paint.values[0] = (double)i / denom;
- remap_color(&cc, pcs, &dcolor, pis, dev,
- gs_color_select_source);
- colors_used |= cmd_drawing_colors_used(cdev, &dcolor);
- }
- }
- }
- }
-
- pie->map_rgb_to_cmyk = dev->color_info.num_components == 4 &&
- base_index == gs_color_space_index_DeviceRGB;
- pie->colors_used.or = colors_used;
- pie->colors_used.slow_rop =
- cmd_slow_rop(dev, pis->log_op, (uses_color ? pdcolor : NULL));
- pie->color_map_is_known = false;
- /*
- * Calculate a (slightly conservative) Y bounding interval for the image
- * in device space.
- */
- {
- int y0 = (int)floor(dbox.p.y - 0.51); /* adjust + rounding slop */
- int y1 = (int)ceil(dbox.q.y + 0.51); /* ditto */
-
- pie->ymin = max(y0, 0);
- pie->ymax = min(y1, dev->height);
- }
-
- /*
- * Make sure the CTM, color space, and clipping region (and, for
- * masked images or images with CombineWithColor, the current color)
- * are known at the time of the begin_image command.
- */
- cmd_clear_known(cdev, clist_image_unknowns(dev, pie) | begin_image_known);
-
- cdev->image_enum_id = pie->id;
- return 0;
-
- /*
- * We couldn't handle the image. Use the default algorithms, which
- * break the image up into rectangles or small pixmaps.
- */
- use_default:
- gs_free_object(mem, pie, "clist_begin_typed_image");
- return gx_default_begin_typed_image(dev, pis, pmat, pic, prect,
- pdcolor, pcpath, mem, pinfo);
- }
-
- /* Process the next piece of an image. */
- private int
- clist_image_plane_data(gx_image_enum_common_t * info,
- const gx_image_plane_t * planes, int yh,
- int *rows_used)
- {
- gx_device *dev = info->dev;
- gx_device_clist_writer * const cdev =
- &((gx_device_clist *)dev)->writer;
- clist_image_enum *pie = (clist_image_enum *) info;
- gs_rect sbox, dbox;
- int y_orig = pie->y;
- int yh_used = min(yh, pie->rect.q.y - y_orig);
- int y0, y1;
- int y, height; /* for BEGIN/END_RECT */
- int code;
-
- #ifdef DEBUG
- if (pie->id != cdev->image_enum_id) {
- lprintf2("end_image id = %lu != clist image id = %lu!\n",
- (ulong) pie->id, (ulong) cdev->image_enum_id);
- *rows_used = 0;
- return_error(gs_error_Fatal);
- }
- #endif
- /****** CAN'T HANDLE VARYING data_x VALUES YET ******/
- {
- int i;
-
- for (i = 1; i < info->num_planes; ++i)
- if (planes[i].data_x != planes[0].data_x) {
- *rows_used = 0;
- return_error(gs_error_rangecheck);
- }
- }
- sbox.p.x = pie->rect.p.x - pie->support.x;
- sbox.p.y = (y0 = y_orig) - pie->support.y;
- sbox.q.x = pie->rect.q.x + pie->support.x;
- sbox.q.y = (y1 = pie->y += yh_used) + pie->support.y;
- gs_bbox_transform(&sbox, &pie->matrix, &dbox);
- /*
- * In order to keep the band list consistent, we must write out
- * the image data in precisely those bands whose begin_image
- * Y range includes the respective image scan lines. Because of
- * rounding, we must expand the dbox by a little extra, and then
- * use image_band_box to calculate the precise range for each band.
- * This is slow, but we don't see any faster way to do it in the
- * general case.
- */
- {
- int ry0 = (int)floor(dbox.p.y) - 2;
- int ry1 = (int)ceil(dbox.q.y) + 2;
- int band_height = cdev->page_band_height;
-
- /*
- * Make sure we don't go into any bands beyond the Y range
- * determined at begin_image time.
- */
- if (ry0 < pie->ymin)
- ry0 = pie->ymin;
- if (ry1 > pie->ymax)
- ry1 = pie->ymax;
- /*
- * If the image extends off the page in the Y direction,
- * we may have ry0 > ry1. Check for this here.
- */
- if (ry0 >= ry1)
- goto done;
- /* Expand the range out to band boundaries. */
- y = ry0 / band_height * band_height;
- height = min(ROUND_UP(ry1, band_height), dev->height) - y;
- }
-
- FOR_RECTS {
- /*
- * Just transmit the subset of the data that intersects this band.
- * Note that y and height always define a complete band.
- */
- gs_int_rect ibox;
- gs_int_rect entire_box;
-
- if (!image_band_box(dev, pie, y, height, &ibox))
- continue;
- /*
- * The transmitted subrectangle has to be computed at the time
- * we write the begin_image command; this in turn controls how
- * much of each scan line we write out.
- */
- {
- int band_ymax = min(band_end, pie->ymax);
- int band_ymin = max(band_end - band_height, pie->ymin);
-
- if (!image_band_box(dev, pie, band_ymin,
- band_ymax - band_ymin, &entire_box))
- continue;
- }
-
- pcls->colors_used.or |= pie->colors_used.or;
- pcls->colors_used.slow_rop |= pie->colors_used.slow_rop;
-
- /* Write out begin_image & its preamble for this band */
- if (!(pcls->known & begin_image_known)) {
- gs_logical_operation_t lop = pie->pis->log_op;
- byte *dp;
- byte *bp = pie->begin_image_command +
- pie->begin_image_command_length;
- uint len;
- byte image_op = cmd_opv_begin_image;
-
- /* Make sure the imager state is up to date. */
- TRY_RECT {
- code = (pie->color_map_is_known ? 0 :
- cmd_put_color_mapping(cdev, pie->pis,
- pie->map_rgb_to_cmyk));
- pie->color_map_is_known = true;
- if (code >= 0) {
- uint want_known = ctm_known | clip_path_known |
- (pie->color_space.id == gs_no_id ? 0 :
- color_space_known);
-
- code = cmd_do_write_unknown(cdev, pcls, want_known);
- }
- if (code >= 0)
- code = cmd_do_enable_clip(cdev, pcls, pie->pcpath != NULL);
- if (code >= 0)
- code = cmd_update_lop(cdev, pcls, lop);
- } HANDLE_RECT(code);
- if (pie->uses_color) {
- TRY_RECT {
- code = cmd_put_drawing_color(cdev, pcls, &pie->dcolor);
- } HANDLE_RECT(code);
- }
- if (entire_box.p.x != 0 || entire_box.p.y != 0 ||
- entire_box.q.x != pie->image.Width ||
- entire_box.q.y != pie->image.Height
- ) {
- image_op = cmd_opv_begin_image_rect;
- cmd_put2w(entire_box.p.x, entire_box.p.y, bp);
- cmd_put2w(pie->image.Width - entire_box.q.x,
- pie->image.Height - entire_box.q.y, bp);
- }
- len = bp - pie->begin_image_command;
- TRY_RECT {
- code =
- set_cmd_put_op(dp, cdev, pcls, image_op, 1 + len);
- } HANDLE_RECT(code);
- memcpy(dp + 1, pie->begin_image_command, len);
-
- /* Mark band's begin_image as known */
- pcls->known |= begin_image_known;
- }
-
- /*
- * The data that we write out must use the X values set by
- * begin_image, which may cover a larger interval than the ones
- * actually needed for these particular scan lines if the image is
- * rotated.
- */
- {
- /*
- * image_band_box ensures that b{x,y}{0,1} fall within
- * pie->rect.
- */
- int bx0 = entire_box.p.x, bx1 = entire_box.q.x;
- int by0 = ibox.p.y, by1 = ibox.q.y;
- int bpp = pie->bits_per_plane;
- int num_planes = pie->num_planes;
- uint offsets[gs_image_max_planes];
- int i, iy, ih, xskip, xoff, nrows;
- uint bytes_per_plane, bytes_per_row, rows_per_cmd;
-
- if (by0 < y0)
- by0 = y0;
- if (by1 > y1)
- by1 = y1;
- /*
- * Make sure we're skipping an integral number of pixels, by
- * truncating the initial X coordinate to the next lower
- * value that is an exact multiple of a byte.
- */
- xoff = bx0 - pie->rect.p.x;
- xskip = xoff & -(int)"\001\010\004\010\002\010\004\010"[bpp & 7];
- for (i = 0; i < num_planes; ++i)
- offsets[i] =
- (by0 - y0) * planes[i].raster + ((xskip * bpp) >> 3);
- bytes_per_plane = ((bx1 - (pie->rect.p.x + xskip)) * bpp + 7) >> 3;
- bytes_per_row = bytes_per_plane * pie->num_planes;
- rows_per_cmd =
- (cbuf_size - cmd_largest_size) / max(bytes_per_row, 1);
-
- if (rows_per_cmd == 0) {
- /* The reader will have to buffer a row separately. */
- rows_per_cmd = 1;
- }
- for (iy = by0, ih = by1 - by0; ih > 0; iy += nrows, ih -= nrows) {
- nrows = min(ih, rows_per_cmd);
- TRY_RECT {
- code = cmd_image_plane_data(cdev, pcls, planes, info,
- bytes_per_plane, offsets,
- xoff - xskip, nrows);
- } HANDLE_RECT(code);
- for (i = 0; i < num_planes; ++i)
- offsets[i] += planes[i].raster * nrows;
- }
- }
- } END_RECTS_ON_ERROR(
- BEGIN
- ++cdev->ignore_lo_mem_warnings;
- NEST_RECT {
- code = write_image_end_all(dev, pie);
- } UNNEST_RECT;
- --cdev->ignore_lo_mem_warnings;
- /* Update sub-rect */
- if (!pie->image.Interpolate)
- pie->rect.p.y += yh_used; /* interpolate & mem recovery currently incompat */
- END,
- (code < 0 ? (band_code = code) : code) >= 0,
- (cmd_clear_known(cdev,
- clist_image_unknowns(dev, pie) | begin_image_known),
- pie->color_map_is_known = false,
- cdev->image_enum_id = pie->id, true)
- );
- done:
- *rows_used = pie->y - y_orig;
- return pie->y >= pie->rect.q.y;
- }
-
- /* Clean up by releasing the buffers. */
- private int
- clist_image_end_image(gx_image_enum_common_t * info, bool draw_last)
- {
- gx_device *dev = info->dev;
- gx_device_clist_writer * const cdev =
- &((gx_device_clist *)dev)->writer;
- clist_image_enum *pie = (clist_image_enum *) info;
- int code;
-
- #ifdef DEBUG
- if (pie->id != cdev->image_enum_id) {
- lprintf2("end_image id = %lu != clist image id = %lu!\n",
- (ulong) pie->id, (ulong) cdev->image_enum_id);
- return_error(gs_error_Fatal);
- }
- #endif
- NEST_RECT {
- do {
- code = write_image_end_all(dev, pie);
- } while (code < 0 && cdev->error_is_retryable &&
- (code = clist_VMerror_recover(cdev, code)) >= 0
- );
- /* if couldn't write successsfully, do a hard flush */
- if (code < 0 && cdev->error_is_retryable) {
- int retry_code;
- ++cdev->ignore_lo_mem_warnings;
- retry_code = write_image_end_all(dev, pie); /* force it out */
- --cdev->ignore_lo_mem_warnings;
- if (retry_code >= 0 && cdev->driver_call_nesting == 0)
- code = clist_VMerror_recover_flush(cdev, code);
- }
- } UNNEST_RECT;
- cdev->image_enum_id = gs_no_id;
- gs_free_object(pie->memory, pie, "clist_image_end_image");
- return code;
- }
-
- /* Create a compositor device. */
- int
- clist_create_compositor(gx_device * dev,
- gx_device ** pcdev, const gs_composite_t * pcte,
- const gs_imager_state * pis, gs_memory_t * mem)
- {
- /****** NYI ******/
- return gx_no_create_compositor(dev, pcdev, pcte, pis, mem);
- }
-
- /* ------ Utilities ------ */
-
- /* Add a command to set data_x. */
- private int
- cmd_put_set_data_x(gx_device_clist_writer * cldev, gx_clist_state * pcls,
- int data_x)
- {
- byte *dp;
- int code;
-
- if (data_x > 0x1f) {
- int dx_msb = data_x >> 5;
-
- code = set_cmd_put_op(dp, cldev, pcls, cmd_opv_set_misc,
- 2 + cmd_size_w(dx_msb));
- if (code >= 0) {
- dp[1] = cmd_set_misc_data_x + 0x20 + (data_x & 0x1f);
- cmd_put_w(dx_msb, dp + 2);
- }
- } else {
- code = set_cmd_put_op(dp, cldev, pcls, cmd_opv_set_misc, 2);
- if (code >= 0)
- dp[1] = cmd_set_misc_data_x + data_x;
- }
- return code;
- }
-
- /* Add commands to represent a halftone order. */
- private int
- cmd_put_ht_order(gx_device_clist_writer * cldev, const gx_ht_order * porder,
- gs_ht_separation_name cname,
- int component /* -1 = default/gray/black screen */ )
- {
- byte command[cmd_max_intsize(sizeof(long)) * 8];
- byte *cp;
- uint len;
- byte *dp;
- uint i, n;
- int code;
- int pi = porder->procs - ht_order_procs_table;
- uint elt_size = porder->procs->bit_data_elt_size;
- const uint nlevels = min((cbuf_size - 2) / sizeof(*porder->levels), 255);
- const uint nbits = min((cbuf_size - 2) / elt_size, 255);
-
- if (pi < 0 || pi > countof(ht_order_procs_table))
- return_error(gs_error_unregistered);
- /* Put out the order parameters. */
- cp = cmd_put_w(component + 1, command);
- if (component >= 0)
- cp = cmd_put_w(cname, cp);
- cp = cmd_put_w(porder->width, cp);
- cp = cmd_put_w(porder->height, cp);
- cp = cmd_put_w(porder->raster, cp);
- cp = cmd_put_w(porder->shift, cp);
- cp = cmd_put_w(porder->num_levels, cp);
- cp = cmd_put_w(porder->num_bits, cp);
- *cp++ = (byte)pi;
- len = cp - command;
- code = set_cmd_put_all_op(dp, cldev, cmd_opv_set_ht_order, len + 1);
- if (code < 0)
- return code;
- memcpy(dp + 1, command, len);
-
- /* Put out the transfer function, if any. */
- code = cmd_put_color_map(cldev, cmd_map_ht_transfer, porder->transfer,
- NULL);
- if (code < 0)
- return code;
-
- /* Put out the levels array. */
- for (i = 0; i < porder->num_levels; i += n) {
- n = porder->num_levels - i;
- if (n > nlevels)
- n = nlevels;
- code = set_cmd_put_all_op(dp, cldev, cmd_opv_set_ht_data,
- 2 + n * sizeof(*porder->levels));
- if (code < 0)
- return code;
- dp[1] = n;
- memcpy(dp + 2, porder->levels + i, n * sizeof(*porder->levels));
- }
-
- /* Put out the bits array. */
- for (i = 0; i < porder->num_bits; i += n) {
- n = porder->num_bits - i;
- if (n > nbits)
- n = nbits;
- code = set_cmd_put_all_op(dp, cldev, cmd_opv_set_ht_data,
- 2 + n * elt_size);
- if (code < 0)
- return code;
- dp[1] = n;
- memcpy(dp + 2, (const byte *)porder->bit_data + i * elt_size,
- n * elt_size);
- }
-
- return 0;
- }
-
- /* Add commands to represent a full (device) halftone. */
- /* We put out the default/gray/black screen last so that the reading */
- /* pass can recognize the end of the halftone. */
- int
- cmd_put_halftone(gx_device_clist_writer * cldev, const gx_device_halftone * pdht,
- gs_halftone_type type)
- {
- uint num_comp = (pdht->components == 0 ? 0 : pdht->num_comp);
-
- {
- byte *dp;
- int code = set_cmd_put_all_op(dp, cldev, cmd_opv_set_misc,
- 2 + cmd_size_w(num_comp));
-
- if (code < 0)
- return code;
- dp[1] = cmd_set_misc_halftone + type;
- cmd_put_w(num_comp, dp + 2);
- }
- if (num_comp == 0)
- return cmd_put_ht_order(cldev, &pdht->order,
- gs_ht_separation_Default, -1);
- {
- int i;
-
- for (i = num_comp; --i >= 0;) {
- int code = cmd_put_ht_order(cldev, &pdht->components[i].corder,
- pdht->components[i].cname, i);
-
- if (code < 0)
- return code;
- }
- }
- return 0;
- }
-
- /* Write out any necessary color mapping data. */
- private int
- cmd_put_color_mapping(gx_device_clist_writer * cldev,
- const gs_imager_state * pis, bool write_rgb_to_cmyk)
- {
- int code;
- const gx_device_halftone *pdht = pis->dev_ht;
-
- /* Put out the halftone. */
- if (pdht->id != cldev->device_halftone_id) {
- code = cmd_put_halftone(cldev, pdht, pis->halftone->type);
- if (code < 0)
- return code;
- cldev->device_halftone_id = pdht->id;
- }
- /* If we need to map RGB to CMYK, put out b.g. and u.c.r. */
- if (write_rgb_to_cmyk) {
- code = cmd_put_color_map(cldev, cmd_map_black_generation,
- pis->black_generation,
- &cldev->black_generation_id);
- if (code < 0)
- return code;
- code = cmd_put_color_map(cldev, cmd_map_undercolor_removal,
- pis->undercolor_removal,
- &cldev->undercolor_removal_id);
- if (code < 0)
- return code;
- }
- /* Now put out the transfer functions. */
- {
- uint which = 0;
- bool all_same = true;
- int i;
-
- for (i = 0; i < countof(cldev->transfer_ids); ++i) {
- if (pis->effective_transfer.indexed[i]->id !=
- cldev->transfer_ids[i]
- )
- which |= 1 << i;
- if (pis->effective_transfer.indexed[i]->id !=
- pis->effective_transfer.indexed[0]->id
- )
- all_same = false;
- }
- /* There are 3 cases for transfer functions: nothing to write, */
- /* a single function, and multiple functions. */
- if (which == 0)
- return 0;
- if (which == (1 << countof(cldev->transfer_ids)) - 1 && all_same) {
- code = cmd_put_color_map(cldev, cmd_map_transfer,
- pis->effective_transfer.indexed[0],
- &cldev->transfer_ids[0]);
- if (code < 0)
- return code;
- for (i = 1; i < countof(cldev->transfer_ids); ++i)
- cldev->transfer_ids[i] = cldev->transfer_ids[0];
- } else
- for (i = 0; i < countof(cldev->transfer_ids); ++i) {
- code = cmd_put_color_map(cldev,
- (cmd_map_index) (cmd_map_transfer_0 + i),
- pis->effective_transfer.indexed[i],
- &cldev->transfer_ids[i]);
- if (code < 0)
- return code;
- }
- }
-
- return 0;
- }
-
- /*
- * Compute the subrectangle of an image that intersects a band;
- * return false if it is empty.
- * It is OK for this to be too large; in fact, with the present
- * algorithm, it will be quite a bit too large if the transformation isn't
- * well-behaved ("well-behaved" meaning either xy = yx = 0 or xx = yy = 0).
- */
- #define I_FLOOR(x) ((int)floor(x))
- #define I_CEIL(x) ((int)ceil(x))
- private void
- box_merge_point(gs_int_rect * pbox, floatp x, floatp y)
- {
- int t;
-
- if ((t = I_FLOOR(x)) < pbox->p.x)
- pbox->p.x = t;
- if ((t = I_CEIL(x)) > pbox->q.x)
- pbox->q.x = t;
- if ((t = I_FLOOR(y)) < pbox->p.y)
- pbox->p.y = t;
- if ((t = I_CEIL(y)) > pbox->q.y)
- pbox->q.y = t;
- }
- private bool
- image_band_box(gx_device * dev, const clist_image_enum * pie, int y, int h,
- gs_int_rect * pbox)
- {
- fixed by0 = int2fixed(y);
- fixed by1 = int2fixed(y + h);
- int
- px = pie->rect.p.x, py = pie->rect.p.y,
- qx = pie->rect.q.x, qy = pie->rect.q.y;
- gs_fixed_rect cbox; /* device clipping box */
- gs_rect bbox; /* cbox intersected with band */
-
- /* Intersect the device clipping box and the band. */
- (*dev_proc(dev, get_clipping_box)) (dev, &cbox);
- /* The fixed_half here is to allow for adjustment. */
- bbox.p.x = fixed2float(cbox.p.x - fixed_half);
- bbox.q.x = fixed2float(cbox.q.x + fixed_half);
- bbox.p.y = fixed2float(max(cbox.p.y, by0) - fixed_half);
- bbox.q.y = fixed2float(min(cbox.q.y, by1) + fixed_half);
- #ifdef DEBUG
- if (gs_debug_c('b')) {
- dlprintf6("[b]band box for (%d,%d),(%d,%d), band (%d,%d) =>\n",
- px, py, qx, qy, y, y + h);
- dlprintf10(" (%g,%g),(%g,%g), matrix=[%g %g %g %g %g %g]\n",
- bbox.p.x, bbox.p.y, bbox.q.x, bbox.q.y,
- pie->matrix.xx, pie->matrix.xy, pie->matrix.yx,
- pie->matrix.yy, pie->matrix.tx, pie->matrix.ty);
- }
- #endif
- if (is_xxyy(&pie->matrix) || is_xyyx(&pie->matrix)) {
- /*
- * The inverse transform of the band is a rectangle aligned with
- * the coordinate axes, so we can just intersect it with the
- * image subrectangle.
- */
- gs_rect ibox; /* bbox transformed back to image space */
-
- if (gs_bbox_transform_inverse(&bbox, &pie->matrix, &ibox) < 0)
- return false;
- pbox->p.x = max(px, I_FLOOR(ibox.p.x));
- pbox->q.x = min(qx, I_CEIL(ibox.q.x));
- pbox->p.y = max(py, I_FLOOR(ibox.p.y));
- pbox->q.y = min(qy, I_CEIL(ibox.q.y));
- } else {
- /*
- * The inverse transform of the band is not aligned with the
- * axes, i.e., is a general parallelogram. To compute an exact
- * bounding box, we need to find the intersections of this
- * parallelogram with the image subrectangle.
- *
- * There is probably a much more efficient way to do this
- * computation, but we don't know what it is.
- */
- gs_point rect[4];
- gs_point corners[5];
- int i;
-
- /* Store the corners of the image rectangle. */
- rect[0].x = rect[3].x = px;
- rect[1].x = rect[2].x = qx;
- rect[0].y = rect[1].y = py;
- rect[2].y = rect[3].y = qy;
- /*
- * Compute the corners of the clipped band in image space. If
- * the matrix is singular or an overflow occurs, the result will
- * be nonsense: in this case, there isn't anything useful we
- * can do, so return an empty intersection.
- */
- if (gs_point_transform_inverse(bbox.p.x, bbox.p.y, &pie->matrix,
- &corners[0]) < 0 ||
- gs_point_transform_inverse(bbox.q.x, bbox.p.y, &pie->matrix,
- &corners[1]) < 0 ||
- gs_point_transform_inverse(bbox.q.x, bbox.q.y, &pie->matrix,
- &corners[2]) < 0 ||
- gs_point_transform_inverse(bbox.p.x, bbox.q.y, &pie->matrix,
- &corners[3]) < 0
- ) {
- if_debug0('b', "[b]can't inverse-transform a band corner!\n");
- return false;
- }
- corners[4] = corners[0];
- pbox->p.x = qx, pbox->p.y = qy;
- pbox->q.x = px, pbox->q.y = py;
- /*
- * We iterate over both the image rectangle and the band
- * parallelogram in a single loop for convenience, even though
- * there is no coupling between the two.
- */
- for (i = 0; i < 4; ++i) {
- gs_point pa, pt;
- double dx, dy;
-
- /* Check the image corner for being inside the band. */
- pa = rect[i];
- gs_point_transform(pa.x, pa.y, &pie->matrix, &pt);
- if (pt.x >= bbox.p.x && pt.x <= bbox.q.x &&
- pt.y >= bbox.p.y && pt.y <= bbox.q.y
- )
- box_merge_point(pbox, pa.x, pa.y);
- /* Check the band corner for being inside the image. */
- pa = corners[i];
- if (pa.x >= px && pa.x <= qx && pa.y >= py && pa.y <= qy)
- box_merge_point(pbox, pa.x, pa.y);
- /* Check for intersections of band edges with image edges. */
- dx = corners[i + 1].x - pa.x;
- dy = corners[i + 1].y - pa.y;
- #define in_range(t, tc, p, q)\
- (0 <= t && t <= 1 && (t = tc) >= p && t <= q)
- if (dx != 0) {
- double t = (px - pa.x) / dx;
-
- if_debug3('b', " (px) t=%g => (%d,%g)\n",
- t, px, pa.y + t * dy);
- if (in_range(t, pa.y + t * dy, py, qy))
- box_merge_point(pbox, (floatp) px, t);
- t = (qx - pa.x) / dx;
- if_debug3('b', " (qx) t=%g => (%d,%g)\n",
- t, qx, pa.y + t * dy);
- if (in_range(t, pa.y + t * dy, py, qy))
- box_merge_point(pbox, (floatp) qx, t);
- }
- if (dy != 0) {
- double t = (py - pa.y) / dy;
-
- if_debug3('b', " (py) t=%g => (%g,%d)\n",
- t, pa.x + t * dx, py);
- if (in_range(t, pa.x + t * dx, px, qx))
- box_merge_point(pbox, t, (floatp) py);
- t = (qy - pa.y) / dy;
- if_debug3('b', " (qy) t=%g => (%g,%d)\n",
- t, pa.x + t * dx, qy);
- if (in_range(t, pa.x + t * dx, px, qx))
- box_merge_point(pbox, t, (floatp) qy);
- }
- #undef in_range
- }
- }
- if_debug4('b', " => (%d,%d),(%d,%d)\n", pbox->p.x, pbox->p.y,
- pbox->q.x, pbox->q.y);
- /*
- * If necessary, add pixels around the edges so we will have
- * enough information to do interpolation.
- */
- if ((pbox->p.x -= pie->support.x) < pie->rect.p.x)
- pbox->p.x = pie->rect.p.x;
- if ((pbox->p.y -= pie->support.y) < pie->rect.p.y)
- pbox->p.y = pie->rect.p.y;
- if ((pbox->q.x += pie->support.x) > pie->rect.q.x)
- pbox->q.x = pie->rect.q.x;
- if ((pbox->q.y += pie->support.y) > pie->rect.q.y)
- pbox->q.y = pie->rect.q.y;
- return (pbox->p.x < pbox->q.x && pbox->p.y < pbox->q.y);
- }
-
- /* Determine which image-related properties are unknown */
- private uint /* mask of unknown properties(see pcls->known) */
- clist_image_unknowns(gx_device *dev, const clist_image_enum *pie)
- {
- gx_device_clist_writer * const cdev =
- &((gx_device_clist *)dev)->writer;
- const gs_imager_state *const pis = pie->pis;
- uint unknown = 0;
-
- /*
- * Determine if the CTM, color space, and clipping region (and, for
- * masked images or images with CombineWithColor, the current color)
- * are unknown. Set the device state in anticipation of the values
- * becoming known.
- */
- if (cdev->imager_state.ctm.xx != pis->ctm.xx ||
- cdev->imager_state.ctm.xy != pis->ctm.xy ||
- cdev->imager_state.ctm.yx != pis->ctm.yx ||
- cdev->imager_state.ctm.yy != pis->ctm.yy ||
- cdev->imager_state.ctm.tx != pis->ctm.tx ||
- cdev->imager_state.ctm.ty != pis->ctm.ty
- ) {
- unknown |= ctm_known;
- cdev->imager_state.ctm = pis->ctm;
- }
- if (pie->color_space.id == gs_no_id) { /* masked image */
- cdev->color_space.space = 0; /* for GC */
- } else { /* not masked */
- if (cdev->color_space.id == pie->color_space.id) {
- /* The color space pointer might not be valid: update it. */
- cdev->color_space.space = pie->color_space.space;
- } else {
- unknown |= color_space_known;
- cdev->color_space = pie->color_space;
- }
- }
- if (cmd_check_clip_path(cdev, pie->pcpath))
- unknown |= clip_path_known;
-
- return unknown;
- }
-
- /* Construct the begin_image command. */
- private int
- begin_image_command(byte *buf, uint buf_size, const gs_image_common_t *pic)
- {
- int i;
- stream s;
- const gs_color_space *ignore_pcs;
- int code;
-
- for (i = 0; i < gx_image_type_table_count; ++i)
- if (gx_image_type_table[i] == pic->type)
- break;
- if (i >= gx_image_type_table_count)
- return_error(gs_error_rangecheck);
- swrite_string(&s, buf, buf_size);
- sputc(&s, (byte)i);
- code = pic->type->sput(pic, &s, &ignore_pcs);
- return (code < 0 ? code : stell(&s));
- }
-
- /* Write data for a partial image. */
- private int
- cmd_image_plane_data(gx_device_clist_writer * cldev, gx_clist_state * pcls,
- const gx_image_plane_t * planes,
- const gx_image_enum_common_t * pie,
- uint bytes_per_plane, const uint * offsets,
- int dx, int h)
- {
- int data_x = planes[0].data_x + dx;
- uint nbytes = bytes_per_plane * pie->num_planes * h;
- uint len = 1 + cmd_size2w(h, bytes_per_plane) + nbytes;
- byte *dp;
- uint offset = 0;
- int plane, i;
- int code;
-
- if (data_x) {
- code = cmd_put_set_data_x(cldev, pcls, data_x);
- if (code < 0)
- return code;
- offset = ((data_x & ~7) * cldev->color_info.depth) >> 3;
- }
- code = set_cmd_put_op(dp, cldev, pcls, cmd_opv_image_data, len);
- if (code < 0)
- return code;
- dp++;
- cmd_put2w(h, bytes_per_plane, dp);
- for (plane = 0; plane < pie->num_planes; ++plane)
- for (i = 0; i < h; ++i) {
- memcpy(dp,
- planes[plane].data + i * planes[plane].raster +
- offsets[plane] + offset,
- bytes_per_plane);
- dp += bytes_per_plane;
- }
- return 0;
- }
-
- /* Write image_end commands into all bands */
- private int /* ret 0 ok, else -ve error status */
- write_image_end_all(gx_device *dev, const clist_image_enum *pie)
- {
- gx_device_clist_writer * const cdev =
- &((gx_device_clist *)dev)->writer;
- int code;
- int y = pie->ymin;
- int height = pie->ymax - y;
-
- /*
- * We need to check specially for images lying entirely outside the
- * page, since FOR_RECTS doesn't do this.
- */
- if (height <= 0)
- return 0;
- FOR_RECTS {
- byte *dp;
-
- if (!(pcls->known & begin_image_known))
- continue;
- TRY_RECT {
- if_debug1('L', "[L]image_end for band %d\n", band);
- code = set_cmd_put_op(dp, cdev, pcls, cmd_opv_image_data, 2);
- } HANDLE_RECT(code);
- dp[1] = 0; /* EOD */
- pcls->known ^= begin_image_known;
- } END_RECTS;
- return 0;
- }
-
- /*
- * Compare a rectangle vs. clip path. Return true if there is no clipping
- * path, if the rectangle is unclipped, or if the clipping path is a
- * rectangle and intersects the given rectangle.
- */
- private bool
- check_rect_for_trivial_clip(
- const gx_clip_path *pcpath, /* May be NULL, clip to evaluate */
- int px, int py, int qx, int qy /* corners of box to test */
- )
- {
- gs_fixed_rect obox;
- gs_fixed_rect imgbox;
-
- if (!pcpath)
- return true;
-
- imgbox.p.x = int2fixed(px);
- imgbox.p.y = int2fixed(py);
- imgbox.q.x = int2fixed(qx);
- imgbox.q.y = int2fixed(qy);
- if (gx_cpath_includes_rectangle(pcpath,
- imgbox.p.x, imgbox.p.y,
- imgbox.q.x, imgbox.q.y))
- return true;
-
- return (gx_cpath_outer_box(pcpath, &obox) /* cpath is rectangle */ &&
- obox.p.x <= imgbox.q.x && obox.q.x >= imgbox.p.x &&
- obox.p.y <= imgbox.q.y && obox.q.y >= imgbox.p.y );
- }
-